summaryrefslogtreecommitdiffstats
path: root/src/android/app/src/main/java/org/citra/citra_emu/utils/ControllerMappingHelper.java
blob: f801a05f0f4a20067b468afca2018cd57e5e2df1 (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
package org.citra.citra_emu.utils;

import android.view.InputDevice;
import android.view.KeyEvent;
import android.view.MotionEvent;

/**
 * Some controllers have incorrect mappings. This class has special-case fixes for them.
 */
public class ControllerMappingHelper {
    /**
     * Some controllers report extra button presses that can be ignored.
     */
    public boolean shouldKeyBeIgnored(InputDevice inputDevice, int keyCode) {
        if (isDualShock4(inputDevice)) {
            // The two analog triggers generate analog motion events as well as a keycode.
            // We always prefer to use the analog values, so throw away the button press
            return keyCode == KeyEvent.KEYCODE_BUTTON_L2 || keyCode == KeyEvent.KEYCODE_BUTTON_R2;
        }
        return false;
    }

    /**
     * Scale an axis to be zero-centered with a proper range.
     */
    public float scaleAxis(InputDevice inputDevice, int axis, float value) {
        if (isDualShock4(inputDevice)) {
            // Android doesn't have correct mappings for this controller's triggers. It reports them
            // as RX & RY, centered at -1.0, and with a range of [-1.0, 1.0]
            // Scale them to properly zero-centered with a range of [0.0, 1.0].
            if (axis == MotionEvent.AXIS_RX || axis == MotionEvent.AXIS_RY) {
                return (value + 1) / 2.0f;
            }
        } else if (isXboxOneWireless(inputDevice)) {
            // Same as the DualShock 4, the mappings are missing.
            if (axis == MotionEvent.AXIS_Z || axis == MotionEvent.AXIS_RZ) {
                return (value + 1) / 2.0f;
            }
            if (axis == MotionEvent.AXIS_GENERIC_1) {
                // This axis is stuck at ~.5. Ignore it.
                return 0.0f;
            }
        } else if (isMogaPro2Hid(inputDevice)) {
            // This controller has a broken axis that reports a constant value. Ignore it.
            if (axis == MotionEvent.AXIS_GENERIC_1) {
                return 0.0f;
            }
        }
        return value;
    }

    private boolean isDualShock4(InputDevice inputDevice) {
        // Sony DualShock 4 controller
        return inputDevice.getVendorId() == 0x54c && inputDevice.getProductId() == 0x9cc;
    }

    private boolean isXboxOneWireless(InputDevice inputDevice) {
        // Microsoft Xbox One controller
        return inputDevice.getVendorId() == 0x45e && inputDevice.getProductId() == 0x2e0;
    }

    private boolean isMogaPro2Hid(InputDevice inputDevice) {
        // Moga Pro 2 HID
        return inputDevice.getVendorId() == 0x20d6 && inputDevice.getProductId() == 0x6271;
    }
}